ModelSim Simulation projects with IP units with and without Quartus Prime Lite

# Contents

| Introduction                                                | 3  |
|-------------------------------------------------------------|----|
| Prerequisites                                               | 3  |
| Lab2_1                                                      | 4  |
| PART1 "Launching ModelSim from Quartus Prime"               |    |
| Objectives                                                  | 4  |
| The project overview                                        | 4  |
| Create a project in Quartus Prime                           | 5  |
| Add IP Components to the design                             | 5  |
| Create units and a top-level design file                    | 10 |
| Setting up ModelSim from Quartus                            | 12 |
| Additional steps and comments                               | 17 |
| Conclusions                                                 | 18 |
| PART2 "Launching ModelSim independently from Quartus Prime" | 19 |
| Objectives                                                  | 19 |
| Launching ModelSim independently from Quartus Prime         | 19 |
| Conclusions                                                 | 21 |

### Introduction

Design simulations are able to include wide range of analyses that virtually test behavior of a product under various operating and environmental conditions.

As opposed to trial-and-error, a smart simulation process allows targeted implementation of design choices in various stages of the development cycle. Proper functional and timing simulation is important to ensure design functionality and success.

This drastically reduces the need for recurrent, time-consuming testing on expensive physical prototypes, and subsequently shortens the total development time.

ModelSim is a multi-language HDL simulation environment offered by Mentor Graphics. *It can be used independently or with Intel Quartus Prime*, which can help create libraries and link the designs to ModelSim. Using the ModelSim Intel FPGA Started Edition software simplifies the debugging with help of simulations.

# Prerequisites

You are expected to:

- be familiar with basics of logic and digital design,
- be familiar with Verilog constructs,
- have Quartus Prime Lite installed.

## Lab2\_1

## PART1 "Launching ModelSim from Quartus Prime"

## **Objectives**

You will use this part to familiarize yourself with the following:

- 1 How to run ModelSim for simulation purposes from Quartus Prime.
- 2 How to use IP functions from IP Catalog of Quartus Prime with ModelSim.
- 3 Some helpful shortcuts for ModelSim.

### The project overviews

This lab utilizes a simple digital logic design shown on Figure 1.



Figure 1

Inputs for the top-level module are:

- clk clock signal (frequency is 25MHz for MiniDilab-CIV board).
- reset reset signal coming from Push Button.

Outputs for the top-level module are:

• LED[7:0] – FPGA pins connected with Leds on a development board.

An algorithm of the project is:

- Input clock is divided by **cnt\_div** unit. Division ratio is a parameter for the unit.
- Unit **cnt\_adr** provides addresses for the **ROM\_8D** unit.
- Unit **ROM\_8D** keeps data and, in accordance of the current address, displays the word (8 bits) of data on the LED[7..0] outputs.
- Reset signal synchronously resets all counters having active value "1".
- Unit **DFF** chain synchronizes external reset signal. It contains two DFF flip-flops connected in a chain.

### Create a project in Quartus Prime

- 1 Start **Quartus Prime Lite** development tool
- 2 Create a project:
  - a. Working directory: C:/Intel\_trn/Q\_MS/lab2\_1
  - b. Project name: lab2\_1
  - c. Top-Level design entity: lab2\_1
  - d. Project Type: Empty project
  - e. Device: EP4CE6E22C8
  - f. EDA tools: ModelSim Altera; Verilog HDL (see Figure 2)



Figure 2

## Add IP Components to the design

1 Create **cnt\_adr** unit by using LPM\_COUNTER.

Unit **cnt\_adr** provides addresses for ROM\_8D unit. You need to create **cnt\_adr** unit during the Lab by using **LPM\_COUNTER** from IP Library.

a. Find LPM\_COUNTER function by typing LPM\_COUNTER in the find field of IP catalog. If you do not have the IP Catalog already open, go to View menu => Utility Windows => IP Catalog to view the window.



Figure 3

- b. Double click LPM\_COUNTER.
- c. Name the file as **cnt\_adr** (*Please make sure to name it as cnt\_adr*) in the **Save IP Variation** box that came up and click **OK**.



Figure 4

d. In the MegaWizard, choose **5 bits** for bus "q" and leave remaining settings to default.



Figure 5

#### Click Next.

e. In the window appeared select Clock enable check box and leave remaining settings to default.



Figure 6

#### Click Next.

f. In the window appeared select synchronous **Clear** check box and leave remaining settings to default.



Figure 7

#### Click Next.

- g. In the window appeared pay attention to the simulation library pointed. It is **lpm** library. When using Native link of Quartus Prime you have all libraries referenced correctly automatically. Leave all settings to default and click **Next**.
- h. In the window appeared select **cnt\_adr\_inst.v** and **cnt\_adr\_bb.v** check boxes and in the working folder you will have templates for using the generated unit in a top level module. Leave the remaining settings to default and click **Finish**.
- i. After you click **Finish** you will get a dialog box asking you if you want to add the IP files generated to the project. Click **Yes**. You will add cnt\_adr.qip file to the project. The file contains information about generated unit, which is for Quartus Prime.

#### 2 Create **ROM 8D** unit by using **ROM: 1-PORT**.

Unit **ROM\_8D** keeps data that are stored in the file **C:\Intel\_trn\Q\_MS\lab2\_1\ ROM\_8D.hex**. You need to create **ROM\_8D** unit during the Lab by using **ROM: 1-PORT** from IP Library.

a. Find **ROM: 1-PORT** function by typing **ROM: 1-PORT** in the find field of IP catalog. If you do not have the IP Catalog already open, go to View menu => Utility Windows => IP Catalog to view the window.



Figure 8

- b. Double click ROM: 1-PORT.
- c. Name the file as **ROM\_8D** (*Please make sure to name it as* **ROM\_8D**) in the **Save IP Variation** box that came up and click **OK**.



Figure 9

d. In the window appeared choose 32 words for the memory. Be sure that the bus "q" is 8 bits wide. Leave the remaining settings to default.



Figure 10

### Click Next.

- e. In the window appeared leave all settings to default. Click **Next**.
- f. In the window appeared point name **ROM\_8D.hex** as the name of the file with memory data. You can type the name directly to the field or use Browse button to select the file from working folder. Leave the remaining settings to default.



Figure 11

#### Click Next.

- g. In the window appeared pay attention to the simulation library pointed. It is **altera\_mf** library. When using Native link of Quartus Prime you have all libraries referenced correctly automatically. Leave all settings to default and click **Next**.
- h. In the window appeared select **ROM\_8D\_inst.v** and **ROM\_8D\_bb.v** check boxes and in the working folder you will have templates for using the generated unit in a top level module. Leave the remaining settings to default and click **Finish**.
- i. After you click **Finish** you will get a dialog box asking you if you want to add the IP files generated to the project. Click **Yes**.
- 3 To check if the IPs are added in the design: select **Project navigator** window => **IP Components** in the drop down menu.



Figure 12

### Create units and a top-level design file

1 You need to create a source code for **DFF\_chain** unit (file name: **DFF\_chain.v**). An example of a code is on Figure 13.

```
1
   module DFF chain
    (input clk,
 3
    input d,
    output reg q );
 5
6
   reg d_int;
7
    always @(posedge clk)
8
   begin
9
        d int
               <= d;
10
        q
               <= d_int;
11
   end
12
13
   endmodule
```

Figure 13

2 You need to create a source code for **cnt\_div** unit (file name: **DFF\_chain.v**). An example of a code is on Figure 14.

```
1 module cnt_div
 2 \#(parameter div = 25)
 3 (input clk,
 4 input reset,
 5 output reg cout );
 7 reg [31:0] cnt;
 8
   wire cycle;
9
   always @(posedge clk)
10
11
       if (reset | cycle)
           cnt <= 0;
12
13
       else
14
      cnt <= cnt+1;</pre>
15
16 assign cycle = (cnt == (div-1));
17
18 always @(posedge clk)
19
       if (reset)
20
           cout <= 1'b0;
       else
21
          if (cycle)
22
23
               cout <= 1'b1;
24
          else
           cout <= 1'b0;
25
26
27 endmodule
```

Figure 14

3 You need to create a source code for top-level unit **lab2\_1** file (file: **lab2\_1.v**).

Unit lab2\_1 is the top level unit, highlighted on Figure 1. It connects all low-level units together. An example of a code is on Figure 15.

```
module lab2 1
 1
 2
      \#(parameter div_by = 25)
 3
      (input
                  CLK,
      input
                  RESET,
 4
 5
      output
                  [7:0] LED );
 6
 7
      wire [4:0] adr;
 8
      wire clk en;
 9
      wire srst;
10
      DFF chain DFF chain inst (
11
                  (CLK
12
          .clk
          .d
13
                  (RESET
14
          .9
                  (srst
15
16
17
      cnt_div #(div_by) cnt_div_inst (
          .clk
                 (CLK
18
          .reset (RESET
19
20
          .cout
                  (clk_en
21
      );
22
23
      cnt_adr cnt_adr_inst (
24
          .clock ( CLK
          .sclr ( RESET
25
26
          .clk_en ( clk_en
27
          .q (adr
28
      );
29
30
      ROM 8D ROM 8D inst (
          .clock ( CLK
32
          .address( adr
33
          .q ( LED
34
      );
35
36
      endmodule
```

Figure 15

- 4 To check if the project (design units description and connections) is correct:
  - a. Select **Project navigator** window => **Hierarchy** in the drop-down menu.
  - b. Select **lab2\_1** top level unit
  - c. Select Processing menu => Start => Start Analysis and Elaboration
  - d. Be sure that you don't have Errors and Critical Warnings.

## Setting up ModelSim from Quartus

1 You need to create a source code for simple testbench **tb Lab2 1** (file: **tb Lab2 1.v**).

```
`timescale 1ns/100ps
 1
 2
     module tb_Lab2_1();
 3
    reg tb clk;
 4
 5
    reg tb reset;
    wire [7:0] tb led;
 7
 8
     Lab2_1 #(5)
 9
     DUT (
10
         . LED
                 (tb led
                              ),
                 (tb clk
11
         .CLK
                              ),
         .RESET (tb reset
12
13
     );
14
     initial
15
     begin
         tb clk
16
                = 1'b0;
         tb_reset = 1'b1;
17
18
         #100 tb reset = 0;
19
         #1000 $stop;
20
21
     end
22
23
     always #10 tb clk = ~tb clk;
24
25
     endmodule
```

Figure 16

This is a simple testbench. Take a look at how the connections are made between the module and the IP.

- First, notice all the inputs of the lab2\_1 module (tb\_clk, tb\_reset) are declared as type reg in the tb\_Lab2\_1.v file.
- Next, all the outputs of the lab2\_1 are declared as type wire i.e. [7:0] tb\_led.
- In the **initial block** (The initial block in Verilog is used generally in test benches. It is a block that only runs once unlike the always block):
  - o all inputs are initialized
  - o Then, after 100 ns, signal **tb reset** switched to inactive state, i.e. "0".
  - o Then after 1000 ns, the simulation process will be stopped by command \$stop.
- always block creates a clock of 50 MHz for CLK input.

Once you have completed the above sections you need to launch ModelSim for simulations. To do so you need to check (and may be to change) settings so that ModelSim can open through Quartus Prime Lite.

- 1 Go to **Tools**  $\rightarrow$  **Options**  $\rightarrow$  **EDA Tool Options**.
- 2 In ModelSim-Altera, enter the executable pathway.

For finding the executable pathway, locate where the Quartus was installed by you (For example: C:\intelFPGA\_lite\20.1\modelsim\_ase\win32aloem).



Figure 17

#### Select **OK** when finished.

- 3 Next step is to make sure that the test bench can be compiled without errors.
  - a. Go to Assignments => Settings => EDA Tool Settings => Simulation.
  - b. Select **ModelSim-Altera** in the **Tool Name** field.
  - c. Under NativeLink Settings select Compile Test Benches and click on Test Benches...
  - d. In the **Test Benches** Dialog Box click on **New...**
  - e. In the field Test bench name, print the name **tb\_Lab2\_1** (It is the top module of the testbench too).
  - f. In the field **File name** select the Testbench file tb\_Lab2\_1.v to be added. Click **Add**.



Figure 18

g. Click **OK**.

- h. Click **OK** again.
- i. Click **OK** again.
- 4 Once all the settings are in place you can go to in the toolbar to **Start Analysis & Synthesis** of the design.
- 5 Next, from Quartus: **Tools=> Run Simulation=> RTL Simulation**
- 6 Once this step is completed. ModelSim should have opened up on the screen. The GUI for ModelSim is shown below.



Figure 19

Note: If you get error loading the design related to source files you might have to go to your working folder, where the project is stored, and check and correct the files with errors.

If you have used ModelSim before and know how to move around the various tabs, you may not see the same layout as shown in Figure 19. You can go to Layout => Reset to get the same layout of the GUI.

#### Workspace:

The workspace consists of various tabs, each tab having a different functionality.

Library tab represents various modules and files present in the design.

Memory List tab will represent the values in the memory units used in the design.

Once the design is simulated, a tab called sim is created which displays the hierarchical structure of the design. You can navigate within the hierarchy by clicking on any line with a '+' (expand) or '-' (contract) icon.

#### Objects Window Pane:

The Objects pane shows the names and current values of data objects in the current region (selected in the Workspace).

Data objects include signals, nets, registers, constants and variables not declared in a process, generics, and parameters.

### Transcript Window:

The Transcript pane provides a command-line interface and serves as an activity log including status and error messages.

#### Text Editor:

This pane allows you to edit and read design files in ModelSim.

#### The Wave window

- 7 Select the tab Wave to see the wave window. You can now observe the waveform and verify the functionality of the design.
  - a. **Undock** the wave window
  - b. **Zoom Full** the wave window .
  - c. Set radix for tb\_led bus as ASCII



Figure 20

You should see A, B, C, D ... letters displayed as on Figure 20.

### Some tips:

• If you want to get to the next rising or falling edge of a particular signal. Select that signal in

the waveform and click on the toolbar for respective actions.

If you want to change the default timescale (for example from ns to us), navigate the cursor on the lower part near the timescale as shown on Figure 21, then right click and select **Grid**, **Timeline & Cursor Control**..



Figure 21

• In the Grid, Timeline & Cursor Control dialogue box, select us as the Time Units as shown on the Figure 22.



Figure 22

Click **OK** and find differences in the waveform window.

- To save this particular set of signals in the waveform:
  - a. Go to **File** => **Save Format** in the wave window.
  - b. Name it "wave\_my\_lab2\_1.do"



Figure 23

c. Click Save.

Next time you, running commands from the Modelsim Transcript, can type "do wave\_my\_lab2\_1.do" before a simulation.

d. The last step: **Stop simulation and close ModelSim window**.

### Additional steps and comments

You can use opened ModelSim as usual ModelSim unless you need to change any IP unit. Therefore, you can change source files and test benches, then recompile ones, restart simulation, debug the source codes. If you need to change IP you need to close ModelSim, switch back to Quartus Prime and change what you want. Then you need to run from Quartus Prime: **Tools=> Run Simulation=> RTL Simulation**.

#### Pay attention:

When you run RTL Simulation from Quartus for the first time, a folder called **simulation** was generated in the Project folder. The folder has sub-folder **Modelsim** – it is the working folder for ModelSim.

- Under **Modelsim** folder look for a .do extension file.

  There is a file "Lab2\_1\_run\_msim\_rtl\_verilog.do" which was created.
- You can run this do file in Modelsim independently from Quartus Prime:
  - Open ModelSim using the start menu.



Figure 24

- Once the software loads:
  - Go to the working folder location, typing in the transcript window the command:

cd C:/Intel\_trn/Q\_MS/lab2\_1/simulation/modelsim

```
# Reading C:/intelFPGA lite/16.1/modelsim ase/tcl/vsim/pref.tcl
ModelSim> cd C:/Intel_trn/Q_MS/lab2_1/simulation/modelsim
# reading modelsim.ini

ModelSim>
```

Figure 25

• Once in the folder, type the command in the transcript window:

```
do Lab2_1_run_msim_rtl_verilog.do
```

The design should load without using the Quartus Software.

- Close the waveform window, if one is open.
- Type in the transcript window the command

do wave\_my\_lab2\_1.do

- Be sure that you have the wave as it was saved early.
- Stop simulation and close ModelSim window.

For more help and shortcuts on Modelsim, you can go to Help  $\rightarrow$  Documentation  $\rightarrow$  Tutorial.

### Conclusions

With this part you learnt how to use IP modules in simulation and to launch ModelSim simulation from Quartus Prime by using Native link.

### PART2 "Launching ModelSim independently from Quartus Prime"

### **Objectives**

You will use this part to familiarize yourself with the following:

How to launching ModelSim independently from Quartus Prime and use IP functions from IP Catalog of Quartus Prime with ModelSim.

## Launching ModelSim independently from Quartus Prime

1 Start ModelSim Intel FPGA Starter Edition.



Figure 26

- 2 Change working Directory.
  - a. Select File => Change Directory and change to the directory C:\Intel trn\Q MS\Lab2 1.
  - b. OR by typing in the Transcript window: cd C:/Intel\_trn/Q\_MS/Lab2\_1
- 3 Create the working library.
  - a. Select **File => New => Library =>work**.
  - b. OR by typing in the Transcript window: **vlib work**
- 4 Compile source files:
  - a. Select **Compile** => **Compile** => "DFF\_chain.v" "ROM\_8D.v" "tb\_Lab2\_1.v" "cnt\_adr.v" "cnt\_div.v" "Lab2\_1.v".
  - b. OR by typing in the Transcript window: vlog -work work "DFF\_chain.v" "ROM\_8D.v" "tb\_Lab2\_1.v" "cnt\_adr.v" "cnt\_div.v" "Lab2\_1.v"



Figure 27

- 5 Be sure that there are no any Errors. If there are ones you will need to correct source files.
- 6 Now you're ready to load the design into the simulator:
  - a. In the Library window, click the '+' sign next to the **work** library to show the files contained there and double-click **tb\_Lab2\_1** to load the design.
  - b. OR by typing in the Transcript window: vsim work.tb\_Lab2\_1
- You will find two errors reported in the Transcript window

```
# Errors: 0, Warnings: 0
ModelSim > vsim work.tb_Lab2_1
# vsim work.tb_Lab2_1
 Start time:
 Loading work.tb Lab2 1
 Loading work.lab2_1
 Loading work.DFF_chain
 Loading work.cnt div
 Loading work.cnt_adr
  ** Error: (vsim-3033) C:/Intel trn/Q MS/lab2 1/cnt adr.v(54): Instantiation of 'lpm counter' failed. The design unit was not found.
    Time: 0 ps Iteration: 0 Instance: /tb_Lab2_1/DUT/cnt_adr_inst File: C:/Intel_trn/Q_MS/lab2_1/cnt_adr.v
          Searched libraries:
             C:/Intel_trn/Q_MS/lab2_1/work
 Loading work.ROM_8D
  ** Error: (vsim-3033) C:/Intel trn/Q MS/lab2 1/ROM 8D.v(59): Instantiation of 'altsyncram' failed. The design unit was not found.
    Time: 0 ps Iteration: 0 Instance: /tb_Lab2_1/DUT/ROM_8D_inst File: C:/Intel_trn/Q_MS/lab2_1/ROM_8D.v
         Searched libraries:
             C:/Intel_trn/Q_MS/lab2_1/work
# Error loading design
                                     Elapsed time: 0:00:00
# End time:
# Errors: 2, Warnings: 0
```

Figure 28

- 8 You need to reference **lpm** and **altera mf** libraries to fix the error.
  - a. Select Simulate => Start Simulation
  - b. In the window appeared the '+' sign next to the work library to show the files contained there and click **tb Lab2 1** to select it.
  - c. Select **Libraries** folder
  - d. In the field Search Libraries click **Add...**
  - e. In the window appeared select **altera mf ver** library



Figure 29

- f. Click OK
- g. Click Add... again
- h. Select **lpm\_ver** library.
- i. Click OK
- j. Click OK
- k. The design will be loaded into the simulator without any errors. If you still have some errors you need to check all steps from the beginning.
- 1. Type in the transcript window the command:

### do C:/Intel\_trn/Q\_MS/lab2\_1/simulation/modelsim/ wave\_my\_lab2\_1.do

Wave window with top-level signals will appear.

- m. Type in the transcript window the command: run 1000 ns
- n. Undock wave window
- o. Zoom full wave window.
- p. Finally, be sure that you see something like the Figure 30.



Figure 30

9 End simulation and close ModelSim window.

### **Conclusions**

With this part you learnt how to use IP modules in simulation and ModelSim simulation independently from Quartus Prime.